# μProcessador 2 Multiplexação, Barramentos, Números: a ULA

### Multiplexação

Um mux é um seletor: ele escolhe uma das entradas possíveis para fornecer à saída.



(Figura: wikipedia)

Se quisermos um mux com 4 entradas ao invés de 2, a descrição fica:

Observe a versão reduzida da tabela verdade do mux 4x1 acima.

| sel1 | sel0 | saida |
|------|------|-------|
| 0    | 0    | entr0 |
| 0    | 1    | entr1 |
| 1    | 0    | entr2 |
| 1    | 1    | entr3 |

Em VHDL há diversas maneiras de fazer multiplexação. Vamos ficar com a seguinte:

Lembre-se: isso não é programação, isso é um circuito! Abaixo vemos o RTL resultante deste trecho.



Mas pra que serve aquele "else '0';" da última linha? Não dá pra retirar?

Na verdade não... Lembre-se que o IEEE 1164 tem *nove* estados lógicos possíveis e *você deve cobrir todos os casos possíveis.* Se levássemos em conta apenas o 'U' (não inicializado) e o 'X' (desconhecido) além de 0 e 1, teremos *quinze* casos, se eu contei direito: 00, 01, 10, 11, 0U, U0, 1U, U1, 0X, X0, 1X, X1, XX, UX e XU. O "else zero" cobre os outros treze deste exemplo, e mais ainda¹.

Por uma esquisitice do VHDL, quando os casos não são totalmente cobertos, ele coloca um registro para os valores, ou seja, vai inserir um *latch* no meio do nosso circuito...(!) Veja abaixo o VHDL, caso você esqueça do "else '0';", e o circuito resultante em RTL.

```
architecture a mux4x1 of mux4x1 is
begin
   saida <=
                entr0
                          when sel1='0' and sel0='0' else
                          when sel1='0' and sel0='1' else
                entr1
                          when sel1='1' and sel0='0' else
                entr2
                          when sel1='1' and sel0='1'; -- OPA! CADÊ O ELSE ZERO?
                entr3
end architecture;
                              saida~6
                                       ATCH ENABLE
                            saida~7
Ou seja:
```

ATENÇÃO: Numa estrutura when-else sempre termine com else '0';

Tá avisado.

Construa um multiplexador 8x1 com as restrições: as entradas 0, 1 e 5 estão sempre em '0', e as entradas 3 e 7 estão sempre em '1'. Portanto, apenas as entradas 2, 4 e 6 terão entradas variáveis, o que dá 3 pinos de dados e 3 pinos de seleção na entidade. Faça um *testbench* adequado.

#### **Barramentos**

Um barramento (bus) é apenas um feixe de sinais, um conjunto de fios. Aqui nesta disciplina vamos usá-los sempre como números inteiros, para facilitar.

Para isso, sempre inclua a biblioteca adicional padrão "numeric\_std" a partir de agora:

```
library ieee;
use ieee.std_logic_1164.all;
use ieee.numeric_std.all; -- para usarmos UNSIGNED
```

Ao definir um barramento, usamos a keyword "unsigned" (como em unsigned integer, inteiro não sinalizado, ou seja, não-negativo) e obrigatoriamente indicamos a largura do barramento em bits.

```
1 É possível simplesmente omitir a condição da última linha, como abaixo: entr2 when sel1='1' and sel0='0' else
```

entr3; A desvantagem é que se houver algum problema, veremos 'entr3' na saída; se ao invés disso observarmos a saída fixa em zero, é mais fácil localizar o *bug*.

2 Há também a opção de usar a biblioteca não padrão "std\_logic\_arith"; mas **não use essa** pois dá conflito e é proprietária.

Por exemplo, vamos fazer um multiplexador 4x1 de largura de 8 bits:

O tipo "unsigned(7 downto 0)" significa que o sinal tem 8 bits (8 pinos) e deve ser tratado como um inteiro não sinalizado, sem números negativos.<sup>3</sup>

A implementação é quase idêntica ao mux de 1 bit:

### Números / Constantes

As constantes devem ser obrigatoriamente representadas por um vetor de bits<sup>4</sup>, representado por aspas duplas. Note que a extensão (*length*) deve bater! Veja exemplos para um sinal "saida" de 8 bits:

```
saida <= "00000010"; -- certo: cte. de 8 bits atribuída a sinal de 8 bits
saida <= "010"; -- pau: mismatch de largura
saida <= 65; -- pau: mismatch de tipo unsigned <= inteiro;</pre>
```

O erro reportado na linha "saida <= 65;" é:

```
porta_e.vhdl:18:13: can't match integer literal with type
array type "unsigned"
ghdl: compilation error
```

pois o VHDL não sabe quantos bits a constante 65 possui<sup>5</sup>. Para resolver de forma simples, use o número equivalente em binário, "01000001", com a largura esperada por "saida".

Será fundamental recortarmos e combinarmos dados. Para isso, usamos o operador de concatenação "&" e o recurso de *slicing*, que obtém um trecho de bits.

Por exemplo, suponha dois barramentos de 8 bits:

```
signal x,y: unsigned(7 downto 0);
```

Podemos combinar os bits dos sinais da forma que quisermos. Como ilustração simples, vamos isolar alguns trechos arbitrários:

- 3 A keyword "downto" indica que a representação é bit-level little endian, ou seja, o bit 0 é o LSB (least significant bit, tem valor numérico 2°). No nosso exemplo, o bit 7 é o MSB (most significant bit) e tem valor 2<sup>7</sup>.
- 4 Números inteiros, como 5 ou 122 por exemplo, são constantes sem especificação de largura em bits e portanto não podem ser usados diretamente com o tipo "unsigned."
- 5 VHDL é uma linguagem *fortemente tipada*, o que na prática significa que ele requer conversões explícitas de tipos por parte do programador. Isso ajuda o código a chegar com menos erros na etapa de simulação.

- (a) O bit mais significativo (MSB) do sinal  $x \in dado por x(7)$
- (b) Os três bits menos significativos (LSB) de x são obtidos com x(2 downto 0)
- (c) Para obter de b5 a b3, basta fazer x(5 downto 3)

Agora se quisermos por bobeira montar um número de 8 bits que começa com '0' e é seguido por (a), (b) e (c) nesta ordem, concatenamos tudo isso com o operador "&":

```
y \le 0^{\circ} \& x(7) \& x(2 \text{ downto } 0) \& x(5 \text{ downto } 3);
```

Portanto se x vale "01101010", y será "00010101". Se x for "11001100", teremos "01100001" em y.

## Operações Aritméticas

Quando usamos o tipo "unsigned"e a biblioteca "numeric\_std", dá pra fazer as paradinhas diretão<sup>6</sup>. Por exemplo, vamos fazer um circuito com duas entradas que calcula tanto a soma quanto a subtração deles simultaneamente:

```
entity soma_e_subtrai is
   port ( x,y : in unsigned(7 downto 0);
        soma, subt : out unsigned(7 downto 0)
   );
end entity;

architecture a_soma_e_subtrai of soma_e_subtrai is
begin
   soma <= x+y;
   subt <= x-y;
end architecture;</pre>
```

Muito mole.

Se quisermos saber se o x é maior do que o y e saber se o x é negativo ou não, incluímos na definição da entidade:

```
maior,x_negativo : out std_logic;
```

E na arquitetura basta colocar as linhas:

Peralá, *negativo?* Sim, dá pra usar números sinalizados de 8 bits pra maioria das coisas (os circuitos são idênticos, pois realizam cálculos em módulo 2<sup>8</sup>). Infelizmente, isso não funciona para a divisão. A comparação acima também *só aceita números positivos*.

Construa um *testbench* para o circuito "soma\_e\_subtrai" e o simule. Isso é importante para você observar corretamente a sintaxe de constantes e compreender a arquitetura acima. **Teste direito:** faça casos relevantes tentando cobrir todas as operações. Por exemplo, se você testou 3+5, não há necessidade de testar 13+15, mas talvez seja interessante testar 100+100 e 200+200... Talvez não precise. Não esqueça de testar negativos, como 18+(-3).

## Tarefa para Entregar: ULA

Nesta primeira entrega quero apenas o arquivo VHDL e o testbench.

Lembre-se que a entrega deste laboratório **vale nota** e não é pouca! Um atraso de poucos dias em um lab não vai impactar tanto a nota, mas *não coma bola!* No conjunto, vale bastante. Consulte as regras da disciplina no Moodle.

Equipes de um ou dois alunos. Aproximadamente a cada dois laboratórios, as equipes deverão apresentar, rápida e pessoalmente, o funcionamento para o professor. Verifique o cronograma.

Especificação da ULA:

- duas entradas de dados de 16 bits;
- uma saída de resultado de 16 bits;
- (opcional) uma ou mais saídas de sinalização de um bit (resultado zero, indicação de "maior" em comparação);
- entradas para seleção de operações.

No mínimo 4 operações, incluindo:

- soma;
- subtração;
- uma maneira de comparar dois números (pode ser operador "maior ou igual", ou então verificar o sinal de um número, ou outra maneira);
- não implemente divisão, pois a implementação do compilador VHDL pode dar uns erros com divisão por zero (são contornáveis, mas melhor evitar).
- Um *testbench* que cobre todas as operações tem que ser entregue; inclua nele um conjunto de testes razoável, que cobre todos os casos de interesse.
- Será utilizada no projeto do processador, portanto escolha operações sensatas.
- Por favor dê nomes claros aos sinais (por ex., op, selec, selec\_op ou selecionaoperacao e não "slo" ou "s")

Resumindo: entram dois dados de 16 bits, escolhe-se qual das 4 operações deve ser executada pela ULA e o resultado desta operação surge na saída de 16 bits. Entregue o .vhd e o \_tb.vhd.

Ou seja, tem que fazer as operações e botar um mux na saída. É assim que se faz.